/*
 * @Author: error: error: git config user.name & please set dead value or install git && error: git config user.email & please set dead value or install git & please set dead value or install git
 * @Date: 2023-10-16 08:57:29
 * @LastEditors: error: error: git config user.name & please set dead value or install git && error: git config user.email & please set dead value or install git & please set dead value or install git
 * @LastEditTime: 2023-11-02 21:55:17
 * @FilePath: /logs/sink.hpp
 * @Description: 这是默认设置,请设置`customMade`, 打开koroFileHeader查看配置 进行设置: https://github.com/OBKoro1/koro1FileHeader/wiki/%E9%85%8D%E7%BD%AE
 */
/*
    日志落地模块的实现
    1.抽象落地基类
    2.派生子类（根据不同的落地方向进行派生）
    3.使用工厂模式进行创建与表示分离
*/
#ifndef __M_SINK_H__
#define __M_SINK_H__
#include "util.hpp"
#include <fstream>
#include <sstream>
#include <memory>
#include <cassert>

namespace zbtlog{
    class LogSink{
        public:
            using ptr = std::shared_ptr<LogSink>;
            LogSink() {}
            virtual ~LogSink()  {}
            virtual void log(const char* data, size_t len) = 0;

    };
    //落地方向：标准输出
    class StdoutSink :public LogSink{
        public:
            //将日志写入到标准输出
            void log(const char* data, size_t len){
                std::cout.write(data, len);
            }
    };
    //落地方向：指定文件
    class FileSink :public LogSink{
        public:
            //构造时传入文件名，并打开文件，将操作句柄管理起来
            FileSink(const std::string &pathname):_pathname(pathname){
                //1.创建日志文件所在的目录
                util::File::createDirectory(util::File::path(pathname));
                //2.创建并打开日志文件
                _ofs.open(_pathname, std::ios::binary | std::ios::app);
                assert(_ofs.is_open());
                
            }
            //将日志写入到标准输出
            void log(const char* data, size_t len){
                _ofs.write(data, len);
                assert(_ofs.good());
            }
        private:
            std::string _pathname;
            std::ofstream _ofs;
    };    
    //落地方向：滚动文件（以大小进行滚动）
    class RollSinkBySize :public LogSink{
        public:
            //构造时传入文件名，并打开文件，将操作句柄管理起来
            RollSinkBySize(const std::string &basename, size_t max_size):
                _basename(basename), _max_fsize(max_size), _cur_fsize(0), _name_cnt(0){
                std::string pathname = createNewFile();
                util::File::createDirectory(util::File::path(pathname));
                //2.创建并打开日志文件
                _ofs.open(pathname, std::ios::binary | std::ios::app);
                assert(_ofs.is_open());               

            }
            //将日志消息写入到标准输出--写如前判断大小，超过最大大小就要切换文件
            void log(const char* data, size_t len){
                if(_cur_fsize >= _max_fsize){
                    _ofs.close();//关闭原来已经打开的文件
                    std::string pathname = createNewFile();
                    _ofs.open(pathname, std::ios::binary | std::ios::app);
                    assert(_ofs.is_open());
                    _cur_fsize = 0;                  
                }
                _ofs.write(data, len);
                assert(_ofs.good());
                _cur_fsize += len;
            }
        private:
            //进行大小判断，超过指定大小则创建新文件
            std::string createNewFile(){
                time_t t = util::Date::now();
                struct tm lt;
                localtime_r(&t, &lt);
                std::stringstream filename;
                filename << _basename;
                filename << lt.tm_year + 1900;
                filename << lt.tm_mon + 1;
                filename << lt.tm_mday;
                filename << lt.tm_hour;
                filename << lt.tm_min;
                filename << lt.tm_sec;
                filename << "-";
                filename << _name_cnt++;
                filename << ".log";
                return filename.str();
            }

        private:
            //基础文件名 + 扩展文件名（以时间生成）组成一个实际的当前输出文件名
            size_t _name_cnt;
            std::string _basename; //.logs/base~ -> .logs/base-200406892342.log
            std::ofstream _ofs;
            size_t _max_fsize;  //记录最大大小，当前文件超过了这个大小就要切换文件
            size_t _cur_fsize;   //记录当前文件已经写入的文件大小
    };

    class SinkFactory{
        public:
            //不定参+完美转发
            template<typename SinkType, typename ...Args>
            static LogSink::ptr create(Args && ...args){
                return std::make_shared<SinkType>(std::forward<Args>(args)...);
            }
    };
    
}
#endif